package com.mmm.station.lang;

import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;

/**
 * 农历日历。<br>
 * 将农历从1901年到2100年之间各年、月的大小以及历年节气保存，然后基于这些数据进行计算。<br>
 * <br>
 * 新增了几个用于农历的常量属性字段，可以使用get()方法获取日历对应的值；<br>
 * 农历年、月、日还可以使用set()/add()/roll()方法设置，其他农历属性自动计算；<br>
 * 另外，还提供了getChinese(int field)方法用于获得农历的中文文字（仅适用于农历属性和星期）。<br>
 * <ul>
 * <li>CHINESE_YEAR - 农历年</li>
 * <li>CHINESE_MONTH - 农历月</li>
 * <li>CHINESE_DATE - 农历日</li>
 * <li>CHINESE_SECTIONAL_TERM - 当月的节气</li>
 * <li>CHINESE_PRINCIPLE_TERM - 当月的中气</li>
 * <li>CHINESE_HEAVENLY_STEM - 农历年的天干</li>
 * <li>CHINESE_EARTHLY_BRANCH - 农历年的地支</li>
 * <li>CHINESE_ZODIAC - 农历年的属相</li>
 * <li>CHINESE_TERM_OR_DATE - 如果当天存在一个节气则指示节气，否则如果当天是初一则指示农历月，否则指示农历日</li>
 * </ul>
 * 注意：<br>
 * 由于Calendar类的设定，公历月份从0起始。所有方法都遵循了这一约定。<br>
 * 但所有的农历属性从1起始。即使是在Calendar提供的方法中，农历月也是从1起始的，并以负数表示闰月。<br>
 * clear()方法在某些情况下会导致农历和公历日期不对应或是不能达到预期的重置效果，应尽量避免使用。<br>
 * 使用getSimpleDateString()获得公历日期字符串时，公历月已经修正；<br>
 * 使用getSimpleChineseDateString()获得农历日期字符串时，农历闰月以*表示。<br>
 * <br>
 * <i>农历算法来源于<a href="http://www.herongyang.com/year_gb/program.html">和荣笔记</a>。</i>
 * @author <a href="http://www.cnblogs.com/huxi/">Huxi</a>
 * @version 0.12 2011-9-5 <br>
 *          <blockquote>修复一个当使用农历正月日期初始化日历时陷入死循环的问题。</blockquote>
 * @version 0.11 2009-12-27 <br>
 *          <blockquote>修复了获取中文农历时未计算农历日期的问题；<br>
 *          加入一个字段CHINESE_TERM_OR_DATE用于模仿台历的显示方式：如果当天有节气则指示节气，如果是初一指示农历月， 否则指示农历日。</blockquote>
 * @version 0.10 2009-12-22
 */
public final class ChineseCalendar extends GregorianCalendar {

	private static final long serialVersionUID = 8L;

	/** 农历年 */
	public static final int CHINESE_YEAR = 801;
	/** 农历月 */
	public static final int CHINESE_MONTH = 802;
	/** 农历日 */
	public static final int CHINESE_DATE = 803;
	/** 当月的节气对应的公历日(前一个节气) */
	public static final int CHINESE_SECTIONAL_TERM = 804;
	/** 当月的节气对应的公历日(后一个节气) */
	public static final int CHINESE_PRINCIPLE_TERM = 805;
	/** 天干 */
	public static final int CHINESE_HEAVENLY_STEM = 806;
	/** 地支 */
	public static final int CHINESE_EARTHLY_BRANCH = 807;
	/** 农历年的属相(生肖) */
	public static final int CHINESE_ZODIAC = 808;
	/** 节气或者农历日 */
	public static final int CHINESE_TERM_OR_DATE = 888;

	private int chineseYear;
	private int chineseMonth; // 1起始，负数表示闰月
	private int chineseDate;
	private int sectionalTerm; // 当月节气的公历日
	private int principleTerm; // 当月中气的公历日

	private boolean areChineseFieldsComputed; // 农历日期是否已经经过计算确认
	private boolean areSolarTermsComputed; // 节气是否已经经过计算确认
	private boolean lastSetChinese; // 最后设置的是不是农历属性

	/** 使用当前时间构造一个实例。 */
	public ChineseCalendar() {
		super();
	}

	/** 使用指定时间构造一个实例。 */
	public ChineseCalendar(Date d) {
		super.setTime(d);
	}

	/** 使用指定时间构造一个实例。 */
	public ChineseCalendar(Calendar c) {
		this(c.getTime());
	}

	/** 使用指定公历日期构造一个实例。 */
	public ChineseCalendar(int y, int m, int d) {
		super(y, m, d);
	}

	/**
	 * 使用指定日期构造一个实例。
	 * @param isChinese 是否为农历日期
	 * @param y
	 * @param m
	 * @param d
	 */
	public ChineseCalendar(boolean isChinese, int y, int m, int d) {
		if (isChinese) {
			set(CHINESE_YEAR, y);
			set(CHINESE_MONTH, m);
			set(CHINESE_DATE, d);
		} else {
			set(y, m, d);
		}
	}

	public void set(int field, int value) {
		computeIfNeed(field);

		if (isChineseField(field)) {
			// 农历属性
			switch (field) {
			case CHINESE_YEAR:
				chineseYear = value;
				break;
			case CHINESE_MONTH:
				chineseMonth = value;
				break;
			case CHINESE_DATE:
				chineseDate = value;
				break;
			default:
				throw new IllegalArgumentException("不支持的field设置：" + field);
			}
			lastSetChinese = true;
		} else {
			// 非农历属性
			super.set(field, value);
			lastSetChinese = false;
		}
		areFieldsSet = false;
		areChineseFieldsComputed = false;
		areSolarTermsComputed = false;
	}

	public int get(int field) {
		computeIfNeed(field);

		if (!isChineseField(field)) {
			return super.get(field);
		}

		switch (field) {
		case CHINESE_YEAR:
			return chineseYear;
		case CHINESE_MONTH:
			return chineseMonth;
		case CHINESE_DATE:
			return chineseDate;
		case CHINESE_SECTIONAL_TERM:
			return sectionalTerm;
		case CHINESE_PRINCIPLE_TERM:
			return principleTerm;
		case CHINESE_HEAVENLY_STEM:
			return (chineseYear - 4) % 10 + 1;
		case CHINESE_EARTHLY_BRANCH:
		case CHINESE_ZODIAC:
			return (chineseYear - 4) % 12 + 1;
		case CHINESE_TERM_OR_DATE:
			int option;
			if (get(Calendar.DATE) == get(CHINESE_SECTIONAL_TERM)) {
				option = CHINESE_SECTIONAL_TERM;
			} else if (get(Calendar.DATE) == get(CHINESE_PRINCIPLE_TERM)) {
				option = CHINESE_PRINCIPLE_TERM;
			} else if (get(CHINESE_DATE) == 1) {
				option = CHINESE_MONTH;
			} else {
				option = CHINESE_DATE;
			}
			return option;
		default:
			throw new IllegalArgumentException("不支持的field获取：" + field);
		}
	}

	public void add(int field, int amount) {
		computeIfNeed(field);

		if (!isChineseField(field)) {
			super.add(field, amount);
			lastSetChinese = false;
			areChineseFieldsComputed = false;
			areSolarTermsComputed = false;
			return;
		}

		switch (field) {
		case CHINESE_YEAR:
			chineseYear += amount;
			break;
		case CHINESE_MONTH:
			for (int i = 0; i < amount; i++) {
				chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
				if (chineseMonth == 1) {
					chineseYear++;
				}
			}
			break;
		case CHINESE_DATE:
			int maxDate = daysInChineseMonth(chineseYear, chineseMonth);
			for (int i = 0; i < amount; i++) {
				chineseDate++;
				if (chineseDate > maxDate) {
					chineseDate = 1;
					chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
					if (chineseMonth == 1) {
						chineseYear++;
					}
					maxDate = daysInChineseMonth(chineseYear, chineseMonth);
				}
			}
		default:
			throw new IllegalArgumentException("不支持的field：" + field);
		}

		lastSetChinese = true;
		areFieldsSet = false;
		areChineseFieldsComputed = false;
		areSolarTermsComputed = false;
	}

	public void roll(int field, int amount) {
		computeIfNeed(field);

		if (!isChineseField(field)) {
			super.roll(field, amount);
			lastSetChinese = false;
			areChineseFieldsComputed = false;
			areSolarTermsComputed = false;
			return;
		}

		switch (field) {
		case CHINESE_YEAR:
			chineseYear += amount;
			break;
		case CHINESE_MONTH:
			for (int i = 0; i < amount; i++) {
				chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
			}
			break;
		case CHINESE_DATE:
			int maxDate = daysInChineseMonth(chineseYear, chineseMonth);
			for (int i = 0; i < amount; i++) {
				chineseDate++;
				if (chineseDate > maxDate) {
					chineseDate = 1;
				}
			}
		default:
			throw new IllegalArgumentException("不支持的field：" + field);
		}

		lastSetChinese = true;
		areFieldsSet = false;
		areChineseFieldsComputed = false;
		areSolarTermsComputed = false;
	}

	/**
	 * 获得属性的中文，可以使用的属性字段为DAY_OF_WEEK以及所有农历属性字段。
	 * @param field
	 * @return
	 */
	public String getChinese(int field) {
		computeIfNeed(field);

		switch (field) {
		case CHINESE_YEAR:
			return getChinese(CHINESE_HEAVENLY_STEM) + getChinese(CHINESE_EARTHLY_BRANCH) + "年";
		case CHINESE_MONTH:
			if (chineseMonth > 0) return chineseMonthNames[chineseMonth] + "月";
			else return "闰" + chineseMonthNames[-chineseMonth] + "月";
		case CHINESE_DATE:
			return chineseDateNames[chineseDate];
		case CHINESE_SECTIONAL_TERM:
			return sectionalTermNames[get(Calendar.MONTH)];
		case CHINESE_PRINCIPLE_TERM:
			return principleTermNames[get(Calendar.MONTH)];
		case CHINESE_HEAVENLY_STEM:
			return stemNames[get(field)];
		case CHINESE_EARTHLY_BRANCH:
			return branchNames[get(field)];
		case CHINESE_ZODIAC:
			return animalNames[get(field)];
		case Calendar.DAY_OF_WEEK:
			return chineseWeekNames[get(field)];
		case CHINESE_TERM_OR_DATE:
			return getChinese(get(CHINESE_TERM_OR_DATE));
		default:
			throw new IllegalArgumentException("不支持的field中文获取：" + field);
		}
	}

	public String getSimpleGregorianDateString() {
		return new StringBuffer().append(get(YEAR)).append("-").append(get(MONTH) + 1).append("-").append(get(DATE)).toString();
	}

	public String getSimpleChineseDateString() {
		return new StringBuffer().append(get(CHINESE_YEAR)).append("-").append(get(CHINESE_MONTH) > 0 ? "" + get(CHINESE_MONTH) : "*" + (-get(CHINESE_MONTH))).append("-").append(get(CHINESE_DATE))
				.toString();
	}

	public String getChineseDateString() {
		return new StringBuffer().append(getChinese(CHINESE_YEAR)).append(getChinese(CHINESE_MONTH)).append(getChinese(CHINESE_DATE)).toString();
	}

	public String toString() {
		StringBuffer buf = new StringBuffer();
		buf.append(getSimpleGregorianDateString()).append(" | ").append(getChinese(DAY_OF_WEEK)).append(" | [农历]").append(getChineseDateString()).append(" ").append(getChinese(CHINESE_ZODIAC))
				.append("年 ").append(get(CHINESE_SECTIONAL_TERM)).append("日").append(getChinese(CHINESE_SECTIONAL_TERM)).append(" ").append(get(CHINESE_PRINCIPLE_TERM)).append("日")
				.append(getChinese(CHINESE_PRINCIPLE_TERM));
		return buf.toString();
	}

	/**
	 * 判断是不是农历属性
	 * @param field
	 * @return
	 */
	private boolean isChineseField(int field) {
		switch (field) {
		case CHINESE_YEAR:
		case CHINESE_MONTH:
		case CHINESE_DATE:
		case CHINESE_SECTIONAL_TERM:
		case CHINESE_PRINCIPLE_TERM:
		case CHINESE_HEAVENLY_STEM:
		case CHINESE_EARTHLY_BRANCH:
		case CHINESE_ZODIAC:
		case CHINESE_TERM_OR_DATE:
			return true;
		default:
			return false;
		}
	}

	/**
	 * 判断是不是与节气有关的属性
	 * @param field
	 * @return
	 */
	private boolean isChineseTermsField(int field) {
		switch (field) {
		case CHINESE_SECTIONAL_TERM:
		case CHINESE_PRINCIPLE_TERM:
		case CHINESE_TERM_OR_DATE:
			return true;
		default:
			return false;
		}
	}

	/**
	 * 如果上一次设置的与这次将要设置或获取的属性不是同一类（农历/公历），<br>
	 * 例如上一次设置的是农历而现在要设置或获取公历，<br>
	 * 则需要先根据之前设置的农历日期计算出公历日期。
	 * @param field
	 */
	private void computeIfNeed(int field) {
		if (isChineseField(field)) {
			if (!lastSetChinese && !areChineseFieldsComputed) {
				super.complete();
				computeChineseFields();
				areFieldsSet = true;
				areChineseFieldsComputed = true;
				areSolarTermsComputed = false;
			}
			if (isChineseTermsField(field) && !areSolarTermsComputed) {
				computeSolarTerms();
				areSolarTermsComputed = true;
			}
		} else {
			if (lastSetChinese && !areFieldsSet) {
				computeGregorianFields();
				super.complete();
				areFieldsSet = true;
				areChineseFieldsComputed = true;
				areSolarTermsComputed = false;
			}
		}
	}

	/**
	 * 使用农历日期计算出公历日期
	 */
	private void computeGregorianFields() {
		int y = chineseYear;
		int m = chineseMonth;
		int d = chineseDate;
		areChineseFieldsComputed = true;
		areFieldsSet = true;
		lastSetChinese = false;

		// 调整日期范围
		if (y < 1900) y = 1899;
		else if (y > 2100) y = 2101;

		if (m < -12) m = -12;
		else if (m > 12) m = 12;

		if (d < 1) d = 1;
		else if (d > 30) d = 30;

		int dateint = y * 10000 + Math.abs(m) * 100 + d;
		if (dateint < 19001111) { // 太小
			set(1901, Calendar.JANUARY, 1);
			super.complete();
		} else if (dateint > 21001201) { // 太大
			set(2100, Calendar.DECEMBER, 31);
			super.complete();
		} else {
			if (Math.abs(m) > 12) {
				m = 12;
			}
			int days = ChineseCalendar.daysInChineseMonth(y, m);
			if (days == 0) {
				m = -m;
				days = ChineseCalendar.daysInChineseMonth(y, m);
			}
			if (d > days) {
				d = days;
			}
			set(y, Math.abs(m) - 1, d);
			computeChineseFields();

			int amount = 0;
			while (chineseYear != y || chineseMonth != m) {
				amount += daysInChineseMonth(chineseYear, chineseMonth);
				chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
				if (chineseMonth == 1) {
					chineseYear++;
				}
			}
			amount += d - chineseDate;

			super.add(Calendar.DATE, amount);
		}
		computeChineseFields();
	}

	/**
	 * 使用公历日期计算出农历日期
	 */
	private void computeChineseFields() {
		int gregorianYear = internalGet(Calendar.YEAR);
		int gregorianMonth = internalGet(Calendar.MONTH) + 1;
		int gregorianDate = internalGet(Calendar.DATE);

		if (gregorianYear < 1901 || gregorianYear > 2100) {
			return;
		}

		int startYear, startMonth, startDate;
		if (gregorianYear < 2000) {
			startYear = baseYear;
			startMonth = baseMonth;
			startDate = baseDate;
			chineseYear = baseChineseYear;
			chineseMonth = baseChineseMonth;
			chineseDate = baseChineseDate;
		} else {
			// 第二个对应日，用以提高计算效率
			// 公历 2000 年 1 月 1 日，对应农历 4697(1999) 年 11 月 25 日
			startYear = baseYear + 99;
			startMonth = 1;
			startDate = 1;
			chineseYear = baseChineseYear + 99;
			chineseMonth = 11;
			chineseDate = 25;
		}

		int daysDiff = 0;

		// 年
		for (int i = startYear; i < gregorianYear; i++) {
			if (isGregorianLeapYear(i)) {
				daysDiff += 366; // leap year
			} else {
				daysDiff += 365;
			}
		}

		// 月
		for (int i = startMonth; i < gregorianMonth; i++) {
			daysDiff += daysInGregorianMonth(gregorianYear, i - 1);
		}

		// 日
		daysDiff += gregorianDate - startDate;

		chineseDate += daysDiff;

		int lastDate = daysInChineseMonth(chineseYear, chineseMonth);
		while (chineseDate > lastDate) {
			chineseDate -= lastDate;
			chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
			if (chineseMonth == 1) {
				chineseYear++;
			}
			lastDate = daysInChineseMonth(chineseYear, chineseMonth);
		}
	}

	/**
	 * 计算节气
	 */
	private void computeSolarTerms() {
		int gregorianYear = internalGet(Calendar.YEAR);
		int gregorianMonth = internalGet(Calendar.MONTH);

		if (gregorianYear < 1901 || gregorianYear > 2100) {
			return;
		}
		sectionalTerm = sectionalTerm(gregorianYear, gregorianMonth);
		principleTerm = principleTerm(gregorianYear, gregorianMonth);
	}

	/* 接下来是静态方法 */
	/**
	 * 是否为公历闰年
	 * @param year
	 * @return
	 */
	public static boolean isGregorianLeapYear(int year) {
		boolean isLeap = false;
		if (year % 4 == 0) {
			isLeap = true;
		}
		if (year % 100 == 0) {
			isLeap = false;
		}
		if (year % 400 == 0) {
			isLeap = true;
		}
		return isLeap;
	}

	/**
	 * 计算公历年当月的天数, 公历月从0起始
	 * @param y
	 * @param m
	 * @return
	 */
	public static int daysInGregorianMonth(int y, int m) {
		int d = daysInGregorianMonth[m];
		if (m == Calendar.FEBRUARY && isGregorianLeapYear(y)) {
			d++; // 公历闰年二月多一天
		}
		return d;
	}

	/**
	 * 计算公历年当月的节气, 公历月从0起始
	 * @param y
	 * @param m
	 * @return
	 */
	public static int sectionalTerm(int y, int m) {
		m++;
		if (y < 1901 || y > 2100) {
			return 0;
		}
		int index = 0;
		int ry = y - baseYear + 1;
		while (ry >= sectionalTermYear[m - 1][index]) {
			index++;
		}
		int term = sectionalTermMap[m - 1][4 * index + ry % 4];
		if ((ry == 121) && (m == 4)) {
			term = 5;
		}
		if ((ry == 132) && (m == 4)) {
			term = 5;
		}
		if ((ry == 194) && (m == 6)) {
			term = 6;
		}
		return term;
	}

	/**
	 * 计算公历年当月的中气, 公历月从0起始
	 * @param y
	 * @param m
	 * @return
	 */
	public static int principleTerm(int y, int m) {
		m++;
		if (y < 1901 || y > 2100) {
			return 0;
		}
		int index = 0;
		int ry = y - baseYear + 1;
		while (ry >= principleTermYear[m - 1][index]) {
			index++;
		}
		int term = principleTermMap[m - 1][4 * index + ry % 4];
		if ((ry == 171) && (m == 3)) {
			term = 21;
		}
		if ((ry == 181) && (m == 5)) {
			term = 21;
		}
		return term;
	}

	/**
	 * 计算农历年的天数
	 * @param y
	 * @param m
	 * @return
	 */
	public static int daysInChineseMonth(int y, int m) {
		// 注意：闰月 m < 0
		int index = y - baseChineseYear + baseIndex;
		int v = 0;
		int l = 0;
		int d = 30;
		if (1 <= m && m <= 8) {
			v = chineseMonths[2 * index];
			l = m - 1;
			if (((v >> l) & 0x01) == 1) {
				d = 29;
			}
		} else if (9 <= m && m <= 12) {
			v = chineseMonths[2 * index + 1];
			l = m - 9;
			if (((v >> l) & 0x01) == 1) {
				d = 29;
			}
		} else {
			v = chineseMonths[2 * index + 1];
			v = (v >> 4) & 0x0F;
			if (v != Math.abs(m)) {
				d = 0;
			} else {
				d = 29;
				for (int i = 0; i < bigLeapMonthYears.length; i++) {
					if (bigLeapMonthYears[i] == index) {
						d = 30;
						break;
					}
				}
			}
		}
		return d;
	}

	/**
	 * 计算农历的下个月
	 * @param y
	 * @param m
	 * @return
	 */
	public static int nextChineseMonth(int y, int m) {
		int n = Math.abs(m) + 1;
		if (m > 0) {
			int index = y - baseChineseYear + baseIndex;
			int v = chineseMonths[2 * index + 1];
			v = (v >> 4) & 0x0F;
			if (v == m) {
				n = -m;
			}
		}
		if (n == 13) {
			n = 1;
		}
		return n;
	}

	/**
	 * @功能: 判断是否为节气
	 * :
	 * @创建日期: 2013-3-21 下午11:34:48
	 * @param chineseDateName
	 * @return
	 */
	public static boolean isTerm(String chineseDateName) {
		for (int i = 0; i < principleTermNames.length; i++) {
			if (principleTermNames[i].equals(chineseDateName)) {
				return true;
			}
		}
		for (int i = 0; i < sectionalTermNames.length; i++) {
			if (sectionalTermNames[i].equals(chineseDateName)) {
				return true;
			}
		}
		return false;
	}

	/* 日历第一天的日期 */
	private static final int baseYear = 1901;
	private static final int baseMonth = 1;
	private static final int baseDate = 1;
	private static final int baseIndex = 0;
	private static final int baseChineseYear = 1900;
	private static final int baseChineseMonth = 11;
	private static final int baseChineseDate = 11;

	/* 中文字符串 */
	private static final String[] chineseWeekNames = { "", "星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六" };
	private static final String[] chineseMonthNames = { "", "正", "二", "三", "四", "五", "六", "七", "八", "九", "十", "十一", "十二" };
	private static final String[] chineseDateNames = { "", "初一", "初二", "初三", "初四", "初五", "初六", "初七", "初八", "初九", "初十", "十一", "十二", "十三", "十四", "十五", "十六", "十七", "十八", "十九", "二十", "廿一", "廿二", "廿三",
			"廿四", "廿五", "廿六", "廿七", "廿八", "廿九", "三十" };
	private static final String[] principleTermNames = { "大寒", "雨水", "春分", "谷雨", "小满", "夏至", "大暑", "处暑", "秋分", "霜降", "小雪", "冬至" };
	private static final String[] sectionalTermNames = { "小寒", "立春", "惊蛰", "清明", "立夏", "芒种", "小暑", "立秋", "白露", "寒露", "立冬", "大雪" };
	private static final String[] stemNames = { "", "甲", "乙", "丙", "丁", "戊", "己", "庚", "辛", "壬", "癸" };
	private static final String[] branchNames = { "", "子", "丑", "寅", "卯", "辰", "巳", "午", "未", "申", "酉", "戌", "亥" };
	private static final String[] animalNames = { "", "鼠", "牛", "虎", "兔", "龙", "蛇", "马", "羊", "猴", "鸡", "狗", "猪" };

	/* 接下来是数据压缩表 */
	private static final int[] bigLeapMonthYears = { 6, 14, 19, 25, 33, 36, 38, 41, 44, 52, 55, 79, 117, 136, 147, 150, 155, 158, 185, 193 };
	private static final char[][] sectionalTermMap = { { 7, 6, 6, 6, 6, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 5, 5, 5, 5, 5, 4, 5, 5 },
			{ 5, 4, 5, 5, 5, 4, 4, 5, 5, 4, 4, 4, 4, 4, 4, 4, 4, 3, 4, 4, 4, 3, 3, 4, 4, 3, 3, 3 }, { 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 5 },
			{ 5, 5, 6, 6, 5, 5, 5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 4, 4, 5, 5, 4, 4, 4, 5, 4, 4, 4, 4, 5 }, { 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 5 },
			{ 6, 6, 7, 7, 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 5 },
			{ 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6, 7, 7, 6, 6, 6, 7, 7 },
			{ 8, 8, 8, 9, 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6, 7, 7, 7 },
			{ 8, 8, 8, 9, 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 7 }, { 9, 9, 9, 9, 8, 9, 9, 9, 8, 8, 9, 9, 8, 8, 8, 9, 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 8 },
			{ 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6, 7, 7, 7 }, { 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6, 7, 7, 6, 6, 6, 7, 7 } };
	private static final char[][] sectionalTermYear = { { 13, 49, 85, 117, 149, 185, 201, 250, 250 }, { 13, 45, 81, 117, 149, 185, 201, 250, 250 }, { 13, 48, 84, 112, 148, 184, 200, 201, 250 },
			{ 13, 45, 76, 108, 140, 172, 200, 201, 250 }, { 13, 44, 72, 104, 132, 168, 200, 201, 250 }, { 5, 33, 68, 96, 124, 152, 188, 200, 201 }, { 29, 57, 85, 120, 148, 176, 200, 201, 250 },
			{ 13, 48, 76, 104, 132, 168, 196, 200, 201 }, { 25, 60, 88, 120, 148, 184, 200, 201, 250 }, { 16, 44, 76, 108, 144, 172, 200, 201, 250 }, { 28, 60, 92, 124, 160, 192, 200, 201, 250 },
			{ 17, 53, 85, 124, 156, 188, 200, 201, 250 } };
	private static final char[][] principleTermMap = { { 21, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 20, 20, 20, 20, 20, 19, 20, 20, 20, 19, 19, 20 },
			{ 20, 19, 19, 20, 20, 19, 19, 19, 19, 19, 19, 19, 19, 18, 19, 19, 19, 18, 18, 19, 19, 18, 18, 18, 18, 18, 18, 18 },
			{ 21, 21, 21, 22, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 21, 20, 20, 20, 20, 19, 20, 20, 20, 20 },
			{ 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 21, 20, 20, 20, 20, 19, 20, 20, 20, 19, 19, 20, 20, 19, 19, 19, 20, 20 },
			{ 21, 22, 22, 22, 21, 21, 22, 22, 21, 21, 21, 22, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 21, 21 },
			{ 22, 22, 22, 22, 21, 22, 22, 22, 21, 21, 22, 22, 21, 21, 21, 22, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 21 },
			{ 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 22, 22, 22, 22, 23 },
			{ 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 23 },
			{ 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 23 },
			{ 24, 24, 24, 24, 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 23 },
			{ 23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 22, 22, 22, 22, 21, 22, 22, 22, 21, 21, 22, 22, 22 },
			{ 22, 22, 23, 23, 22, 22, 22, 23, 22, 22, 22, 22, 21, 22, 22, 22, 21, 21, 22, 22, 21, 21, 21, 22, 21, 21, 21, 21, 22 } };
	private static final char[][] principleTermYear = { { 13, 45, 81, 113, 149, 185, 201 }, { 21, 57, 93, 125, 161, 193, 201 }, { 21, 56, 88, 120, 152, 188, 200, 201 },
			{ 21, 49, 81, 116, 144, 176, 200, 201 }, { 17, 49, 77, 112, 140, 168, 200, 201 }, { 28, 60, 88, 116, 148, 180, 200, 201 }, { 25, 53, 84, 112, 144, 172, 200, 201 },
			{ 29, 57, 89, 120, 148, 180, 200, 201 }, { 17, 45, 73, 108, 140, 168, 200, 201 }, { 28, 60, 92, 124, 160, 192, 200, 201 }, { 16, 44, 80, 112, 148, 180, 200, 201 },
			{ 17, 53, 88, 120, 156, 188, 200, 201 } };

	private static final char[] daysInGregorianMonth = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
	private static final char[] chineseMonths = { 0x00, 0x04, 0xad, 0x08, 0x5a, 0x01, 0xd5, 0x54, 0xb4, 0x09, 0x64, 0x05, 0x59, 0x45, 0x95, 0x0a, 0xa6, 0x04, 0x55, 0x24, 0xad, 0x08, 0x5a, 0x62, 0xda,
			0x04, 0xb4, 0x05, 0xb4, 0x55, 0x52, 0x0d, 0x94, 0x0a, 0x4a, 0x2a, 0x56, 0x02, 0x6d, 0x71, 0x6d, 0x01, 0xda, 0x02, 0xd2, 0x52, 0xa9, 0x05, 0x49, 0x0d, 0x2a, 0x45, 0x2b, 0x09, 0x56, 0x01,
			0xb5, 0x20, 0x6d, 0x01, 0x59, 0x69, 0xd4, 0x0a, 0xa8, 0x05, 0xa9, 0x56, 0xa5, 0x04, 0x2b, 0x09, 0x9e, 0x38, 0xb6, 0x08, 0xec, 0x74, 0x6c, 0x05, 0xd4, 0x0a, 0xe4, 0x6a, 0x52, 0x05, 0x95,
			0x0a, 0x5a, 0x42, 0x5b, 0x04, 0xb6, 0x04, 0xb4, 0x22, 0x6a, 0x05, 0x52, 0x75, 0xc9, 0x0a, 0x52, 0x05, 0x35, 0x55, 0x4d, 0x0a, 0x5a, 0x02, 0x5d, 0x31, 0xb5, 0x02, 0x6a, 0x8a, 0x68, 0x05,
			0xa9, 0x0a, 0x8a, 0x6a, 0x2a, 0x05, 0x2d, 0x09, 0xaa, 0x48, 0x5a, 0x01, 0xb5, 0x09, 0xb0, 0x39, 0x64, 0x05, 0x25, 0x75, 0x95, 0x0a, 0x96, 0x04, 0x4d, 0x54, 0xad, 0x04, 0xda, 0x04, 0xd4,
			0x44, 0xb4, 0x05, 0x54, 0x85, 0x52, 0x0d, 0x92, 0x0a, 0x56, 0x6a, 0x56, 0x02, 0x6d, 0x02, 0x6a, 0x41, 0xda, 0x02, 0xb2, 0xa1, 0xa9, 0x05, 0x49, 0x0d, 0x0a, 0x6d, 0x2a, 0x09, 0x56, 0x01,
			0xad, 0x50, 0x6d, 0x01, 0xd9, 0x02, 0xd1, 0x3a, 0xa8, 0x05, 0x29, 0x85, 0xa5, 0x0c, 0x2a, 0x09, 0x96, 0x54, 0xb6, 0x08, 0x6c, 0x09, 0x64, 0x45, 0xd4, 0x0a, 0xa4, 0x05, 0x51, 0x25, 0x95,
			0x0a, 0x2a, 0x72, 0x5b, 0x04, 0xb6, 0x04, 0xac, 0x52, 0x6a, 0x05, 0xd2, 0x0a, 0xa2, 0x4a, 0x4a, 0x05, 0x55, 0x94, 0x2d, 0x0a, 0x5a, 0x02, 0x75, 0x61, 0xb5, 0x02, 0x6a, 0x03, 0x61, 0x45,
			0xa9, 0x0a, 0x4a, 0x05, 0x25, 0x25, 0x2d, 0x09, 0x9a, 0x68, 0xda, 0x08, 0xb4, 0x09, 0xa8, 0x59, 0x54, 0x03, 0xa5, 0x0a, 0x91, 0x3a, 0x96, 0x04, 0xad, 0xb0, 0xad, 0x04, 0xda, 0x04, 0xf4,
			0x62, 0xb4, 0x05, 0x54, 0x0b, 0x44, 0x5d, 0x52, 0x0a, 0x95, 0x04, 0x55, 0x22, 0x6d, 0x02, 0x5a, 0x71, 0xda, 0x02, 0xaa, 0x05, 0xb2, 0x55, 0x49, 0x0b, 0x4a, 0x0a, 0x2d, 0x39, 0x36, 0x01,
			0x6d, 0x80, 0x6d, 0x01, 0xd9, 0x02, 0xe9, 0x6a, 0xa8, 0x05, 0x29, 0x0b, 0x9a, 0x4c, 0xaa, 0x08, 0xb6, 0x08, 0xb4, 0x38, 0x6c, 0x09, 0x54, 0x75, 0xd4, 0x0a, 0xa4, 0x05, 0x45, 0x55, 0x95,
			0x0a, 0x9a, 0x04, 0x55, 0x44, 0xb5, 0x04, 0x6a, 0x82, 0x6a, 0x05, 0xd2, 0x0a, 0x92, 0x6a, 0x4a, 0x05, 0x55, 0x0a, 0x2a, 0x4a, 0x5a, 0x02, 0xb5, 0x02, 0xb2, 0x31, 0x69, 0x03, 0x31, 0x73,
			0xa9, 0x0a, 0x4a, 0x05, 0x2d, 0x55, 0x2d, 0x09, 0x5a, 0x01, 0xd5, 0x48, 0xb4, 0x09, 0x68, 0x89, 0x54, 0x0b, 0xa4, 0x0a, 0xa5, 0x6a, 0x95, 0x04, 0xad, 0x08, 0x6a, 0x44, 0xda, 0x04, 0x74,
			0x05, 0xb0, 0x25, 0x54, 0x03 };

}
